package com.awg.shiro.realm;

import com.awg.dao.IUserDao;
import com.awg.vo.User;
import org.apache.shiro.authc.AuthenticationException;
import org.apache.shiro.authc.AuthenticationInfo;
import org.apache.shiro.authc.AuthenticationToken;
import org.apache.shiro.authc.SimpleAuthenticationInfo;
import org.apache.shiro.authz.AuthorizationInfo;
import org.apache.shiro.authz.SimpleAuthorizationInfo;
import org.apache.shiro.crypto.hash.Md5Hash;
import org.apache.shiro.realm.AuthorizingRealm;
import org.apache.shiro.subject.PrincipalCollection;
import org.apache.shiro.util.ByteSource;
import org.springframework.util.ObjectUtils;
import javax.annotation.Resource;
import java.util.Set;

/**
 * 自定义Realm
 *
 * @author awg
 * @date 2020-02-21
 * @version 1.0
 */
public class CustomRealm extends AuthorizingRealm {

    @Resource
    private IUserDao userDaoImpl;

    // 模拟生成加密后的信息
    public static void main(String[] args) {
        // 使用Md5加盐并加密2次
        // Hello123     132world        05637e4f92f7e280a8bdafb4f00f3544
        // u320Jk       Ixs^3%Mxs&      c17972a3fde6d2c505cbe7f0b534cd84
        // myuser33     1323%Muscix     409a70a47acc74a30103775207b8f289
        Md5Hash md5Hash = new Md5Hash("1323%Muscix", "myuser33", 2);
        System.out.println(md5Hash.toString());
    }

    /**
     * 认证
     * @param token 主体传递过来的认证信息
     * @return 认证信息
     * @throws AuthenticationException 认证出现错误的异常
     */
    @Override
    protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken token) throws AuthenticationException {
        // 从主体传递过来的认证信息中获取用户名, principal(主要的)
        String username = (String) token.getPrincipal();
        // 通过用户名到数据库中获取凭证（密码）
        String password = getPasswordByUserName(username);
        // 判断符合 用户名-密码 的用户是否存在
        // 若不存在，则返回null
        if (password == null) {
            return null;
        }
        // 若存在，则返回一个AuthenticationInfo的对象
        // principal: 是与指定领域相关联的“主要”主体, 名字随便取。
        // realmName: 自定义的Realm的名字,随便取
        SimpleAuthenticationInfo authenticationInfo = new SimpleAuthenticationInfo(username, password, "customRealm");
        // 设置"盐", "盐"一般使用随机数的方式产生
        authenticationInfo.setCredentialsSalt(ByteSource.Util.bytes(username));
        return authenticationInfo;
    }

    /**
     * 授权
     * @param principals 主体信息
     * @return 授权信息
     */
    @Override
    protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principals) {
        // 从主体认证信息中获取用户名
        String userName = (String) principals.getPrimaryPrincipal();
        // 通过用户名从数据库或者缓存中获取用户的角色信息
        Set<String> roles = getRolesByUserName(userName);
        // 通过用户名从数据库或者缓存中获取用户的角色信息所对应的权限信息
        Set<String> permissions = getPermissionsByUserName(userName);
        // 将角色、权限设置给主体授权信息
        SimpleAuthorizationInfo authorizationInfo = new SimpleAuthorizationInfo();
        authorizationInfo.setRoles(roles);
        authorizationInfo.setStringPermissions(permissions);
        return authorizationInfo;
    }

    /**
     * 通过用户名从数据库或者缓存中获取用户的角色信息所对应的权限信息
     * @param username 用户名
     * @return 通过用户名在数据库或者缓存中获得的用户的角色信息所对应的权限信息
     */
    private Set<String> getPermissionsByUserName(String username) {
        Set<String> permissionsSet = userDaoImpl.findPermissionsByUsername(username);
        if (ObjectUtils.isEmpty(permissionsSet)) {
            return null;
        }
        return permissionsSet;
    }

    /**
     * 通过用户名从数据库或者缓存中获取用户的角色信息
     * @param username 用户名
     * @return 通过用户名在数据库或者缓存中获得的用户的角色信息
     */
    private Set<String> getRolesByUserName(String username) {
        Set<String> rolesSet = userDaoImpl.findRolesByUsername(username);
        if (ObjectUtils.isEmpty(rolesSet)) {
            return null;
        }
        return rolesSet;
    }

    /**
     * 通过用户名到数据库中获取凭证（密码）
     * @param username 用户名
     * @return 通过用户名在数据库中获得的凭证（密码）
     */
    private String getPasswordByUserName(String username) {
        User user = userDaoImpl.getUserByUserName(username);
        if (ObjectUtils.isEmpty(user)) {
            return null;
        }
        return user.getPassword();
    }

}
